// MIT License
// 
// Copyright (c) 2018 Cong Feng.
// 
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
// 
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
// 
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.

// -*- Mode: c++; c-basic-offset: 4; tab-width: 4; -*-

/******************************************************************************
 *
 *  file:  ArgException.h
 *
 *  Copyright (c) 2003, Michael E. Smoot .
 *  All rights reserved.
 *
 *  See the file COPYING in the top directory of this distribution for
 *  more information.
 *
 *  THE SOFTWARE IS PROVIDED _AS IS_, WITHOUT WARRANTY OF ANY KIND, EXPRESS
 *  OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 *  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
 *  THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 *  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
 *  FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
 *  DEALINGS IN THE SOFTWARE.
 *
 *****************************************************************************/

#ifndef TCLAP_ARG_EXCEPTION_H
#define TCLAP_ARG_EXCEPTION_H

#include <exception>
#include <string>

namespace TCLAP {

/**
 * A simple class that defines and argument exception.  Should be caught
 * whenever a CmdLine is created and parsed.
 */
class ArgException : public std::exception {
public:
  /**
   * Constructor.
   * \param text - The text of the exception.
   * \param id - The text identifying the argument source.
   * \param td - Text describing the type of ArgException it is.
   * of the exception.
   */
  ArgException(const std::string &text = "undefined exception",
               const std::string &id = "undefined",
               const std::string &td = "Generic ArgException")
      : std::exception(), _errorText(text), _argId(id), _typeDescription(td) {}

  /**
   * Destructor.
   */
  virtual ~ArgException() throw() {}

  /**
   * Returns the error text.
   */
  std::string error() const { return (_errorText); }

  /**
   * Returns the argument id.
   */
  std::string argId() const {
    if (_argId == "undefined")
      return " ";
    else
      return ("Argument: " + _argId);
  }

  /**
   * Returns the arg id and error text.
   */
  const char *what() const throw() {
    static std::string ex;
    ex = _argId + " -- " + _errorText;
    return ex.c_str();
  }

  /**
   * Returns the type of the exception.  Used to explain and distinguish
   * between different child exceptions.
   */
  std::string typeDescription() const { return _typeDescription; }

private:
  /**
   * The text of the exception message.
   */
  std::string _errorText;

  /**
   * The argument related to this exception.
   */
  std::string _argId;

  /**
   * Describes the type of the exception.  Used to distinguish
   * between different child exceptions.
   */
  std::string _typeDescription;
};

/**
 * Thrown from within the child Arg classes when it fails to properly
 * parse the argument it has been passed.
 */
class ArgParseException : public ArgException {
public:
  /**
   * Constructor.
   * \param text - The text of the exception.
   * \param id - The text identifying the argument source
   * of the exception.
   */
  ArgParseException(const std::string &text = "undefined exception",
                    const std::string &id = "undefined")
      : ArgException(text, id,
                     std::string("Exception found while parsing ") +
                         std::string("the value the Arg has been passed.")) {}
};

/**
 * Thrown from CmdLine when the arguments on the command line are not
 * properly specified, e.g. too many arguments, required argument missing, etc.
 */
class CmdLineParseException : public ArgException {
public:
  /**
   * Constructor.
   * \param text - The text of the exception.
   * \param id - The text identifying the argument source
   * of the exception.
   */
  CmdLineParseException(const std::string &text = "undefined exception",
                        const std::string &id = "undefined")
      : ArgException(text, id,
                     std::string("Exception found when the values ") +
                         std::string("on the command line do not meet ") +
                         std::string("the requirements of the defined ") +
                         std::string("Args.")) {}
};

/**
 * Thrown from Arg and CmdLine when an Arg is improperly specified, e.g.
 * same flag as another Arg, same name, etc.
 */
class SpecificationException : public ArgException {
public:
  /**
   * Constructor.
   * \param text - The text of the exception.
   * \param id - The text identifying the argument source
   * of the exception.
   */
  SpecificationException(const std::string &text = "undefined exception",
                         const std::string &id = "undefined")
      : ArgException(text, id,
                     std::string("Exception found when an Arg object ") +
                         std::string("is improperly defined by the ") +
                         std::string("developer.")) {}
};

class ExitException {
public:
  ExitException(int estat) : _estat(estat) {}

  int getExitStatus() const { return _estat; }

private:
  int _estat;
};

} // namespace TCLAP

#endif